UI 界面传值

传值需求

  • 将用户信息 userInfo 作为传值对象进行传递。

场景一 主页传值到详情页

  • 现在模拟传递用户名:userName。
属性传值
  1. 属性传值一般用于从主页传值到详情页。
  2. 传值步骤:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
steps 1:在DetailViewController.h文件中将需要获取的值声明成属性。
#import <UIKit/UIKit.h>

@interface DetailViewController : UIViewController

@property (nonatomic, strong)NSString *userInfo; /**< 用户信息 */

@end

steps 2:在HomeViewController.m文件中导入头文件“DetailViewController.h”,然后在界面跳转逻辑处理方法中初始化DetailViewController,并通过点语法给属性userInfo赋需要传递的值。
- (void)respondsToButton:(UIButton *)sender {

// 初始化详情视图控制器n
DetailViewController *detailVc = [[DetailViewController alloc] init];

// 属性传值:赋值
NSDictionary *userInfo = @{@"name":@"Charles", @"age":@(22)};
detailVc.userInfo = userInfo;

// 模态切换(界面跳转)
[self presentViewController:detailVc animated:YES completion:nil];

}

steps 3:在DetailViewController.m文件viewDidLoad方法中获取userName的值,此时获取到的值就是从主页传过来的值。
- (void)viewDidLoad {
[super viewDidLoad];

NSLog(@"%@", userInfo);
}
init传值
  1. init方法传值与属性传值类似,一般用于从主页传值到详情页。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
steps 1:在DetailViewController.h文件中声明init方法。
- (instancetype)initWithUserInfo:(NSDictionary *)userInfo; /**< init传值方法声明 */
steps 2:在HomeViewController.m文件中导入头文件“DetailViewController.h”,然后在界面跳转逻辑处理方法中通过initWithUserInfo:方法初始化DetailViewController并赋值。
- (void)respondsToButton:(UIButton *)sender {

NSDictionary *userInfo = @{@"name":@"Charles", @"age":@(22)};

// 初始化详情视图控制器
DetailViewController *detailVc = [[DetailViewController alloc] initWithUserInfo:userInfo];

// 模态切换(界面跳转)
[self presentViewController:detailVc animated:YES completion:nil];

}
steps 3:在DetailViewController.m文件中重写init方法,即实现initWithUserInfo:方法,在这个方法中获取userName的值,此时获取到的值就是从主页传过来的值。
- (instancetype)initWithUserInfo:(NSDictionary *)userInfo {
if (self = [super init]) {
NSLog(@"%@", userInfo);
}
return self;
}

场景二 详情页传值到主页

Block块传值
  • block在传值中主要用于回调,现模拟从详情视图控制器传值到主页视图控制器。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
steps 1:在 DetailViewController.h文件中声明block类型、属性以及block回调方法。
#import <UIKit/UIKit.h>

// 1 声明block类型

typedef void(^CallBackBlock)(NSString *context);

@interface DetailViewController : UIViewController

// 2 声明block属性

@property (nonatomic, copy) CallBackBlock callBackBlock;

/ 3 声明block传值方法
- (void)getsUserInfoWithBlocks:(CallBackBlock)callBackBlock;


@end
steps 2:在 DetailViewController.m文件中,实现如下操作:
// 4 赋值属性block
- (void)getsUserInfoWithBlocks:(CallBackBlock)callBackBlock {
self.callBackBlock = callBackBlock;
}

// 处理按钮点击
- (void)respondsToButtonClick:(UIButton *)sender {
// 5 传值
if (self.callBack) {

NSDictionary *userInfo = @{@"name":@"Charles", @"age":@(22)};
self.callBackBlock(userInfo);

[self dismissViewControllerAnimated:YES completion:nil];
}
}
steps 3:在 ViewController.m文件实现如下操作:
- (void)respondsToButtonClick:(UIButton *)sender {
DetailViewController *detailVc = [[DetailViewController alloc] init];

// 6 调用block,取值
[detailVc getsUserInfoWithBlocks:^(NSDictionary *userInfo) {
NSLog(@"%@", userInfo);
}];

// 模态切换(界面跳转)
[self presentViewController:detailVc animated:YES completion:nil];
}
  • Tips:
    1、为block取别名,可在参数列表中将需要传递的参数写成形参;
    2、设置block属性注意使用copy关键字;
    3、设置一个方法持有当前block;
    4、在合适的地方进行调用类似于代理;
    5、在创建该对象的地方进行block方面的调用;
协议传值
  1. 协议传值又称代理传值,可直接将需要传递的值从委托方传送至代理人,协议传值可用于从下一个视图控制器传值到上一个视图控制器(详情页传值到主页),现假定主页是详情页的代理。
  • 传值步骤
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
steps 1:在DetailViewController.h文件中声明协议,并且设置代理属性。
#import <UIKit/UIKit.h>


// @class 意在告诉编译器,“DetailViewController”为一个类。

@class DetailViewController;

// @protocol 声明协议
// 协议命名规范:类名 + delegate

@protocol DetailViewControllerDelegate <NSObject>

// @optional:声明可选协议方法
// 协议方法的声明模仿苹果官方声明方式,将类实例以及传递信息一并暴露在参数中

@optional

- (void)detailViewController:(DetailViewController *)detailViewController goBackWithUserInfo:(NSDictionary *)userInfo;

@end

@interface DetailViewController : UIViewController

// 声明代理属性,注意关键字使用 weak || assign,可避免保留环
@property (nonatomic, weak) id <DetailViewControllerDelegate> delegate;
@end
steps 2:在DetailViewController.m文件处理返回按钮方法中调用协议方法传值。
- (void)respondsToButton:(UIButton *)sender {

// 首先判断代理人是否存在并且是否遵守协议并且实现了协议方法

if (_delegate && [_delegate respondsToSelector:@selector(detailViewController:goBackWithUserInfo:)]) {

// 如果满足判断条件,则让代理执行协议方法,此处让代理人执行协议方法,在代理人那个控制器中的协议方法会被执行;

// 通常经协议传值在此处调用方法时,直接给参数赋值即可,在代理人控制器实现的协议方法中,可直接获取此处设置的值;

NSDictionary *userInfo = @{@"name":@"Charles", @"age":@(22)};

[_delegate detailViewController:self goBackWithUserInfo:userInfo];

}
}
steps 3:在HomeViewController.m文件处理界面跳转按钮方法中初始化详情视图控制器,设置详情视图控制器协议代理为self(主页),并且遵守<DetailViewControllerDelegate>协议。
- (void)respondsToButton:(UIButton *)sender {
// 初始化详情视图控制器
DetailViewController *detailVc = [[DetailViewController alloc] init];

// 设置代理,并且遵守<DetailViewControllerDelegate>
detailVc.delegate = self;

// 模态切换(界面跳转)
[self presentViewController:detailVc animated:YES completion:nil];
}
steps 4:在HomeViewController.m中实现<DetailViewControllerDelegate>协议方法,获取值。
#pragma mark *** DetailViewControllerDelegate ***
// 实现协议方法,获取值
- (void)detailViewController:(DetailViewController *)detailViewController goBackWithUserInfo:(NSDictionary *)userInfo {
NSLog(@"%@", userInfo);
}

场景三 多界面传值

通知传值
  1. 通知传值适用于任意控制器(界面),不管两个控制器之间是否有关联,只需满足一个条件,在传值的时候必须保证通知已经被设定,即已添加通知(观察者observer)。现假设从详情界面传值到主界面,即从下一个界面传值到上一个界面,具体实现方式如下。
  • 传值步骤
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
steps 1:注册通知:为保证在传值时通知已经被设定,因此需要在HomeViewController.m文件中注册通知。
#import "HomeViewController.h"
#import "DetailViewController.h"

@interface HomeViewController ()

@end

@implementation HomeViewController

- (instancetype)init {
self = [super init];
if (self) {

/**
* 注册通知
*
* @param observer 观察者对象
* @param selector 触发方法,即当收到通知之后执行的方法
* @param name 通知代号,即通知标识,发送通知时的标识必须和注册通知时的标识一致
* @param object 是否传值,在注册通知的时候无需值,因此此处可填nil
*
*/

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(respondsToNotification:) name:@"notification_name" object:nil];
}
return self;
}

- (void)viewDidLoad {
[super viewDidLoad];
}

#pragma mark *** responds notification ***

// 处理通知,当接收到通知的时候该方法会自动调用
// 在此处获取从发送通知的控制器传过来的值
- (void)respondsToNotification:(NSNotification *)info {

}
steps 2:发送通知:在DetailViewController.m文件处理返回按钮方法中,发送通知,传值到主界面,发送通知时的标识必须与注册通知时的标识一致。
- (void)respondsToButton:(UIButton *)sender {

// 发送通知:通知标识必须与注册通知时的标识一致
// 将需要传递的信息以字典形式赋给 userInfo 参数
NSDictionary *userInfo = @{@"name":@"Charles", @"age":@(22)};
[[NSNotificationCenter defaultCenter] postNotificationName:@"notification_name" object:nil userInfo:userInfo];

}
steps 3:处理通知:在HomeViewController.m文件处理通知方法中,获取值。
#pragma mark *** responds notification ***

// 处理通知,在此处获取从发送通知的控制器传过来的值
// 注意:info参数包含两个属性,可通过点语法访问。
// 1、name:为对应通知的标识
// 2、userInfo:为传递的信息
- (void)respondsToNotification:(NSNotification *)info {
NSLog(@"%@", info.userInfo);
}
steps 4:移除通知:通知在界面被释放的时候一定记得移除,否则可能会导致程序的奔溃。移除通知在注册通知控制器中的[dealloc]方法中实现。
- (void)dealloc
{
// 移除通知
[[NSNotificationCenter defaultCenter] removeObserver:self];
}
  • 注意
    1、通知传值的使用会贯穿如下4个步骤:注册通知 -> 发送通知 -> 处理通知 -> 移除通知
    2、通知必须先注册再使用,通知必须在不需要的时候调用remove方法移除。
单例传值
  1. 单例贯穿整个应用程序声明周期,利用单例传值适用于任何控制器,使用前提是在获取值的时候必须保证单例属性有值,否则获取值为nil,此处模拟从主页视图控制器传值到详情视图控制器。
  • 传值步骤
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
steps 1:创建单例,继承于NSObject,任意命名,必须符合规范。此处创建单例类名为Singleton。
steps 2:在Singleton.h中声明传值属性,并且声明单例类便利构造器。
@interface Singleton : NSObject

@property (nonatomic, strong) NSDictionary *userInfo; /**< 单例属性 */

+ (instancetype)defaultSingleton; /**< 单例便利构造器 */

@end
steps 3:在Singleton.m文件中实现遍历构造器方法。
#import "Singleton.h"

static Singleton *singleton = nil;

@implementation Singleton

+ (instancetype)defaultSingleton {

// GCD创建单例,效率更高,性能更好,消耗更低。
static dispatch_once_t onceToken;
dispatch_once(&onceToken, ^{
singleton = [[Singleton alloc] init];
});
return singleton;
}
@end
steps 4:在HomeViewController.m文件中获取单例实例,并且赋值单例属性,赋值位置可根据实际情况进行调整。
// 单例属性赋值
- (void)viewDidLoad {
[super viewDidLoad];

// 获取单例实例,首先需导入Singleton.h
Singleton *singleton = [Singleton defaultSingleton];

// 单例属性赋值
NSDictionary *userInfo = @{@"name":@"Charles", @"age":@(22)};
singleton.userInfo = userInfo;
}
steps 5:在DetailViewController.m文件中获取单例属性,取值位置可根据实际情况进行调整。
// 获取单例属性
- (void)viewDidLoad {
[super viewDidLoad];

// 获取单例实例,首先需导入Singleton.h
Singleton *singleton = [Singleton defaultSingleton];

// 获取单例属性值
NSLog(@"%@", singleton.userInfo);
}
NSUserDefaults传值
  1. NSUserDefaults系统单例传值和自定义单例传值基本一致,首先需保证NSUserDefaults对应key中有值,此处模拟主页视图控制器传值到详情视图控制器
  • 传值步骤
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
steps 1:在HomeViewController.m中获取NSUserDefaults实例,并且存值。
- (void)saveValueInUserDefaults {

// 获取NSUserDefaults实例
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];

// 存值
NSDictionary *userInfo = @{@"name":@"Charles", @"age":@(22)};
[defaults setObject:userInfo forKey:@"userInfo"];

// 同步数据
[defaults synchronize];
}
steps 2:在DetailViewController.m中获取值
- (void)getValueInUserDefaults {

// 获取NSUserDefaults实例
NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];

// 根据key获取值
NSDictionary *userInfo = [defaults objectForKey:@"userInfo"];

NSLog(@"%@", userInfo);
}